]> git.saurik.com Git - apple/libdispatch.git/blobdiff - examples/Dispatch Samples/proc.c
libdispatch-187.5.tar.gz
[apple/libdispatch.git] / examples / Dispatch Samples / proc.c
diff --git a/examples/Dispatch Samples/proc.c b/examples/Dispatch Samples/proc.c
deleted file mode 100644 (file)
index 511b42f..0000000
+++ /dev/null
@@ -1,209 +0,0 @@
-/*
- * Copyright (c) 2008 Apple Inc.  All rights reserved.
- *
- * @APPLE_DTS_LICENSE_HEADER_START@
- * 
- * IMPORTANT:  This Apple software is supplied to you by Apple Computer, Inc.
- * ("Apple") in consideration of your agreement to the following terms, and your
- * use, installation, modification or redistribution of this Apple software
- * constitutes acceptance of these terms.  If you do not agree with these terms,
- * please do not use, install, modify or redistribute this Apple software.
- * 
- * In consideration of your agreement to abide by the following terms, and
- * subject to these terms, Apple grants you a personal, non-exclusive license,
- * under Apple's copyrights in this original Apple software (the "Apple Software"),
- * to use, reproduce, modify and redistribute the Apple Software, with or without
- * modifications, in source and/or binary forms; provided that if you redistribute
- * the Apple Software in its entirety and without modifications, you must retain
- * this notice and the following text and disclaimers in all such redistributions
- * of the Apple Software.  Neither the name, trademarks, service marks or logos of
- * Apple Computer, Inc. may be used to endorse or promote products derived from
- * the Apple Software without specific prior written permission from Apple.  Except
- * as expressly stated in this notice, no other rights or licenses, express or
- * implied, are granted by Apple herein, including but not limited to any patent
- * rights that may be infringed by your derivative works or by other works in
- * which the Apple Software may be incorporated.
- * 
- * The Apple Software is provided by Apple on an "AS IS" basis.  APPLE MAKES NO
- * WARRANTIES, EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION THE IMPLIED
- * WARRANTIES OF NON-INFRINGEMENT, MERCHANTABILITY AND FITNESS FOR A PARTICULAR
- * PURPOSE, REGARDING THE APPLE SOFTWARE OR ITS USE AND OPERATION ALONE OR IN
- * COMBINATION WITH YOUR PRODUCTS. 
- * 
- * IN NO EVENT SHALL APPLE BE LIABLE FOR ANY SPECIAL, INDIRECT, INCIDENTAL OR
- * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
- * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * ARISING IN ANY WAY OUT OF THE USE, REPRODUCTION, MODIFICATION AND/OR
- * DISTRIBUTION OF THE APPLE SOFTWARE, HOWEVER CAUSED AND WHETHER UNDER THEORY OF
- * CONTRACT, TORT (INCLUDING NEGLIGENCE), STRICT LIABILITY OR OTHERWISE, EVEN IF
- * APPLE HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- * 
- * @APPLE_DTS_LICENSE_HEADER_END@
- */
-
-#include <stdio.h>
-#include <string.h>
-#include <strings.h>
-#include <stdarg.h>
-#include <assert.h>
-#include <stdlib.h>
-#include <malloc/malloc.h>
-#include <sys/stat.h>
-#include <sys/mman.h>
-#include <fcntl.h>
-#include <unistd.h>
-#include <spawn.h>
-#include <errno.h>
-
-#include <libkern/OSAtomic.h>
-
-#include <dispatch/dispatch.h>
-
-extern char **environ;
-dispatch_queue_t qpf;
-volatile int exitcount = 0;
-
-// maximum value for exitcount before we quit
-#define proccount  2
-
-
-struct qp_msg {
-    FILE *f;
-    char *str;
-};
-
-void qpf_puts(void *m_) {
-    struct qp_msg *m = m_;
-    fputs(m->str, m->f);
-    free(m->str);
-    free(m);
-}
-
-void qfprintf(FILE *f, const char *fmt, ...) {
-    va_list ap;
-    va_start(ap, fmt);
-    struct qp_msg *m = malloc(sizeof(struct qp_msg));
-    assert(m);
-    vasprintf(&m->str, fmt, ap);
-    m->f = f;
-    dispatch_async(qpf, ^(void) { qpf_puts(m); });
-    va_end(ap);
-}
-
-#define qprintf(fmt...) qfprintf(stdout, ## fmt)
-
-/* context structure, contains a process id and the 
- * command line arguments used to launch it. Used to 
- * provide context info to the block associated
- * with a process event source.
- */
-struct pinfo {
-    pid_t pid;
-       dispatch_source_t source;
-    char **argv;
-};
-
-/* pid_finalize() is called when the dispatch source is released.
- * this block is attached to the attribute that is passed to dispatch_source_proc_create(),
- * and is thus associated with the dispatch source. */
-void pid_finalize(struct pinfo *pi) {
-       qprintf("process %d is done watching %s (%d)\n", getpid(), pi->argv[0], pi->pid);
-       dispatch_release(pi->source);
-       if (OSAtomicIncrement32(&exitcount) == proccount) {
-               qprintf("both processes exited\n");
-               dispatch_sync(qpf,^{});
-               exit(0);
-       }
-}
-
-
-/* pid_event() is called from a block that is associated with a process event
- * source for a specific process id (via dispatch_source_proc_create()). When
- * such an event occurs, pid_event() calls dispatch_source_get_context() to
- * gain access to the pid and process name that were stored in the context at
- * the time the block was attached to the event source.
- */
-#define FLAG(X) ((dispatch_source_get_data(src) & DISPATCH_PROC_##X) ? #X" " : "")
-
-void pid_event(struct pinfo *pi) {
-       dispatch_source_t src = pi->source;
-
-       qprintf("process %d %s, flags: %x %s%s%s%s\n", pi->pid, pi->argv[0], dispatch_source_get_data(src), FLAG(EXIT), FLAG(FORK), FLAG(EXEC), FLAG(SIGNAL));
-       if (dispatch_source_get_data(src) & DISPATCH_PROC_EXIT) {
-               int s;
-               waitpid(dispatch_source_get_handle(src), &s, WNOHANG);
-               qprintf("  %s exit status %d\n", pi->argv[0], s);
-               dispatch_source_cancel(src);
-       }
-}
-
-/* proc_start() takes a context pointer (ppi), and a dispatch queue (pq),
- * and spawns the process named in ppi->argv[0]. The resulting process id
- * is stored in the context (ppi->pid). On successfully spawning the process,
- * it creates a dispatch source for the purpose of executing the routine pid_event(pi,ev)
- * when certain events (exit, fork, exec, reap, or signal) occur to the process.
- */
-void proc_start(void *ppi, dispatch_queue_t pq) {
-    struct pinfo *pi = ppi;
-       
-    int rc = posix_spawnp(&pi->pid, pi->argv[0], NULL, NULL, pi->argv, environ);
-    if (rc) {
-               int e = errno;
-               qprintf("Can't spawn %s (rc=%d, e=%d %s)\n", pi->argv[0], rc, e, strerror(e));
-    } else {
-
-               dispatch_source_t dsp = dispatch_source_create(DISPATCH_SOURCE_TYPE_PROC, pi->pid, DISPATCH_PROC_EXIT|DISPATCH_PROC_FORK|DISPATCH_PROC_EXEC|DISPATCH_PROC_SIGNAL, pq);
-               dispatch_source_set_event_handler_f(dsp, (dispatch_function_t)pid_event);
-               dispatch_source_set_cancel_handler_f(dsp,  (dispatch_function_t)pid_finalize);
-               pi->source = dsp;
-               dispatch_set_context(dsp, pi);
-               dispatch_resume(dsp);
-               
-               qprintf("process %d spawned %s: %d, watching with event source: %p\n", getpid(), pi->argv[0], pi->pid, dsp);
-               
-    }
-}
-
-int main(int argc, char *argv[]) {
-    struct pinfo pi, pi2, pi3;
-       struct pinfo  *ppi2 = & pi2, *ppi3 = &pi3;
-       
-    char *av[] = {argv[0], NULL}; // set up context info (struct pinfo) for this process.
-    pi.pid = getpid();
-    pi.argv = av;
-       
-    char *av2[] = {"sleep", "3", NULL};        // set up context info (struct pinfo) for the sleep tool
-    pi2.argv = av2;
-       
-    char *av3[] = {"script", "/tmp/LOG", "banner", "-w80", "!", NULL}; // set up context info (struct pinfo) for the script tool
-    pi3.argv = av3;
-       
-    dispatch_queue_t pq = dispatch_queue_create("PQ", NULL);   // create our main processing queue
-       
-    qpf = dispatch_queue_create("qprintf", NULL);                              // create a separate queue for printf
-               
-       /* create a dispatch source that will call the routine pid_event(pi,ev)
-        * when certain events occur to the specified process (pi->pid). The dispatch source is
-        * associated with the dispatch queue that was created in this routine (pq). This example
-        * requests the block be executed whenever one of the following events occurs:
-        *              exit, fork, exec, reap, or signal.
-        */
-       dispatch_source_t procSource = dispatch_source_create(DISPATCH_SOURCE_TYPE_PROC, pi.pid, DISPATCH_PROC_EXIT|DISPATCH_PROC_FORK|DISPATCH_PROC_EXEC|DISPATCH_PROC_SIGNAL, pq);
-       
-       dispatch_source_set_event_handler_f(procSource, (dispatch_function_t)pid_event);
-       dispatch_source_set_cancel_handler_f(procSource,  (dispatch_function_t)pid_finalize);
-       pi.source = procSource;
-       dispatch_set_context(procSource, &pi);
-       dispatch_resume(procSource);
-       
-       /* create a block (which simply calls proc_start()), and dispatch it to the queue.
-        * proc_start() will spawn the process named by ppiX->argv[0], and set up
-        * another block (containing a call to pid_event()) on an event source that
-        * will recieve process events...
-        */
-       dispatch_async(pq, ^(void) { proc_start( ppi2, pq ); });        // launch the sleep tool, and create the process watcher for it
-       dispatch_async(pq, ^(void) { proc_start( ppi3, pq ); });        // launch the script tool, and create the process watcher for it
-       
-       
-    dispatch_main();                                                                                   // wait for all the queued and spawned items to finish...
-}